#pragma once

#include <optional>

#include <server/http/request_handler_base.hpp>
#include <userver/components/component_context.hpp>
#include <userver/engine/mutex.hpp>
#include <userver/engine/task/task_processor_fwd.hpp>
#include <userver/engine/task/task_with_result.hpp>
#include <userver/server/handlers/handler_base.hpp>
#include <userver/server/http/http_request.hpp>
#include <userver/utils/statistics/metrics_storage.hpp>
#include <userver/utils/token_bucket.hpp>

#include "handler_info_index.hpp"

USERVER_NAMESPACE_BEGIN

namespace server::http {

class HttpRequestHandler final : public RequestHandlerBase {
public:
    HttpRequestHandler(
        const components::ComponentContext& component_context,
        const std::optional<std::string>& logger_access_component,
        const std::optional<std::string>& logger_access_tskv_component,
        bool is_monitor,
        std::string server_name
    );

    using NewRequestHook = std::function<void(std::shared_ptr<http::HttpRequest>)>;
    void SetNewRequestHook(NewRequestHook hook);

    engine::TaskWithResult<void> StartRequestTask(std::shared_ptr<http::HttpRequest> request) const override;

    void DisableAddHandler();
    void AddHandler(const handlers::HttpHandlerBase& handler, engine::TaskProcessor& task_processor);
    bool IsAddHandlerDisabled() const noexcept;
    const HandlerInfoIndex& GetHandlerInfoIndex() const override;

    const logging::TextLoggerPtr& LoggerAccess() const noexcept override { return logger_access_; }
    const logging::TextLoggerPtr& LoggerAccessTskv() const noexcept override { return logger_access_tskv_; }

    void SetRpsRatelimit(std::optional<size_t> rps);

    void SetRpsRatelimitStatusCode(HttpStatus status_code);

private:
    engine::TaskWithResult<void> StartFailsafeTask(std::shared_ptr<http::HttpRequest> http_request) const;

    logging::TextLoggerPtr logger_access_;
    logging::TextLoggerPtr logger_access_tskv_;

    HandlerInfoIndex handler_info_index_;

    const bool is_monitor_;
    const std::string server_name_;
    NewRequestHook new_request_hook_;
    mutable utils::TokenBucket rate_limit_;
    std::atomic<HttpStatus> cc_status_code_{HttpStatus::kTooManyRequests};
    std::chrono::steady_clock::time_point cc_enabled_tp_;
    utils::statistics::MetricsStoragePtr metrics_;
    dynamic_config::Source config_source_;
};

}  // namespace server::http

USERVER_NAMESPACE_END
